package org.example.entity;

import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArraySet;

@Component
@Slf4j
@ServerEndpoint("/websocketTest/{userId}")  // 接口路径 ws://localhost:8087/webSocketTest/userId;
public class WebSocketTest {
    //与某个客户端的连接会话，需要通过它来给客户端发送数据
    private Session session;

    /**
     * 用户ID
     */
    private String userId;
    //concurrent包的线程安全Set，用来存放每个客户端对应的MyWebSocket对象。
    //虽然@Component默认是单例模式的，但springboot还是会为每个websocket连接初始化一个bean，所以可以用一个静态set保存起来。
    //  注：底下WebSocket是当前类名
    // 这两个是static，所有连接的对象都会共用这两个变量，所以才能存储所有连接的信息
    // 而每个连接都会单独创建一个WebSocketTest对象，即这是单例模式
    private static CopyOnWriteArraySet<WebSocketTest> webSockets =new CopyOnWriteArraySet<>();
    // 用来存在线连接用户信息
    private static ConcurrentHashMap<String,Session> sessionPool = new ConcurrentHashMap<String,Session>();
    /**
     * 链接成功调用的方法
     */
    @OnOpen
    public void onOpen(Session session, @PathParam(value="userId")String userId) {
        try {
            this.session = session;
            this.userId = userId;
            webSockets.add(this);
            sessionPool.put(userId, session);
            log.info("【websocket消息】有新的连接,userId:"+userId+"，总数为:"+webSockets.size());
            // 发送连接成功消息(Result格式)

        } catch (Exception e) {
            log.error("WebSocket链接调用错误");
        }
    }

    /**
     * 链接关闭调用的方法
     */
    @OnClose
    public void onClose() {
        try {
            webSockets.remove(this);
            sessionPool.remove(this.userId);
            log.info("【websocket消息】连接断开，总数为:"+webSockets.size());
        } catch (Exception e) {
            log.error("WebSocket链接断开错误");
        }
    }

    /**
     * 收到客户端消息后调用的方法
     *
     * @param message
     *
     */
    @OnMessage
    public void onMessage(String message) { // message是json格式数据
        log.info("【websocket消息】收到客户端消息:"+message);
        // 测试发送消息功能
        sendOneMessage(this.userId,"Message Received!");
    }
    /**
     * 定时任务 （测试时再打开）
     */
//    @Scheduled(cron = "0/3 * * * * ?  ")
//    public void refreshOnlineDevices(){
//        if(!sessionPool.isEmpty()){
//            SimpleDateFormat formatter= new SimpleDateFormat("yyyy-MM-dd 'at' HH:mm:ss z");
//            Date date = new Date(System.currentTimeMillis());
////            System.out.println("scheduled msg to "+this.userId+" "+formatter.format(date));
////            sendOneMessage(this.userId,"test to "+this.userId+" "+formatter.format(date));
//            sendAllMessage("test to All "+formatter.format(date));
//        } else {
//            System.out.println("No Connection!");
//        }
//    }

    /** 发送错误时的处理
     * @param session
     * @param error
     */
    @OnError
    public void onError(Session session, Throwable error) {

        log.error("用户错误,原因:"+error.getMessage());
        // 发送错误信息
        error.printStackTrace();
    }

    // 此为广播消息
    public void sendAllMessage(String message) {
        log.info("【websocket消息】广播消息:"+message);
        for(WebSocketTest webSocket : webSockets) {
            try {
                if(webSocket.session.isOpen()) {
                    webSocket.session.getAsyncRemote().sendText(message);
                }
            } catch (Exception e) {
                log.error("广播消息错误");
                e.printStackTrace();
            }
        }
    }

    // 此为单点消息
    public void sendOneMessage(String userId, String message) {
        Session session = sessionPool.get(userId);
        if (session != null&&session.isOpen()) {
            try {
                log.info("【websocket消息】 单点消息:"+message);
                session.getAsyncRemote().sendText(message);
            } catch (Exception e) {
                log.error("单点消息错误, customerTitle="+userId);
                e.printStackTrace();
            }
        }
    }

    // 此为单点消息(多人)
    public void sendMoreMessage(String[] userIds, String message) {
        for(String userId:userIds) {
            sendOneMessage(userId,message);
        }
    }
}
